}
static void
-gtk_widget_invalidate_paintable_contents (GtkWidget *widget)
+gtk_widget_update_paintables (GtkWidget *widget)
{
GtkWidgetPrivate *priv = gtk_widget_get_instance_private (widget);
GSList *l;
- if (!_gtk_widget_is_drawable (widget))
- return;
-
for (l = priv->paintables; l; l = l->next)
- gtk_widget_paintable_invalidate_contents (l->data);
-}
-
-static void
-gtk_widget_invalidate_paintable_size (GtkWidget *widget)
-{
- GtkWidgetPrivate *priv = gtk_widget_get_instance_private (widget);
- GSList *l;
-
- for (l = priv->paintables; l; l = l->next)
- gtk_widget_paintable_invalidate_size (l->data);
+ gtk_widget_paintable_update_image (l->data);
}
/**
priv->draw_needed = TRUE;
g_clear_pointer (&priv->render_node, gsk_render_node_unref);
- gtk_widget_invalidate_paintable_contents (widget);
if (_gtk_widget_get_has_surface (widget) &&
_gtk_widget_get_realized (widget))
gdk_surface_queue_expose (gtk_widget_get_surface (widget));
}
}
- gtk_widget_invalidate_paintable_size (widget);
-
if (_gtk_widget_get_visible (widget))
{
GtkWidget *parent = _gtk_widget_get_parent (widget);
priv->alloc_needed = FALSE;
priv->alloc_needed_on_child = FALSE;
- gtk_widget_invalidate_paintable_size (widget);
+ gtk_widget_update_paintables (widget);
check_clip:
if (size_changed || baseline_changed)
priv->allocation.height = 0;
memset (&priv->allocated_size, 0, sizeof (priv->allocated_size));
priv->allocated_size_baseline = 0;
- gtk_widget_invalidate_paintable_size (widget);
+ gtk_widget_update_paintables (widget);
}
}
gtk_widget_unmap (child);
}
- gtk_widget_invalidate_paintable_contents (widget);
+ gtk_widget_update_paintables (widget);
gtk_widget_unset_state_flags (widget,
GTK_STATE_FLAG_PRELIGHT |
priv->render_node = render_node;
priv->draw_needed = FALSE;
+
+ gtk_widget_update_paintables (widget);
}
if (priv->render_node)
#include "gtkintl.h"
#include "gtksnapshot.h"
+#include "gtkrendernodepaintableprivate.h"
#include "gtkwidgetprivate.h"
/**
GtkWidget *widget;
guint loop_tracker;
- guint size_invalid : 1;
- guint contents_invalid : 1;
+ GdkPaintable *current_image; /* the image that we are presenting */
+ GdkPaintable *pending_image; /* the image that we should be presenting */
+ guint pending_update_cb; /* the idle source that updates the valid image to be the new current image */
};
struct _GtkWidgetPaintableClass
double height)
{
GtkWidgetPaintable *self = GTK_WIDGET_PAINTABLE (paintable);
- graphene_matrix_t transform;
- self->contents_invalid = FALSE;
-
- if (self->widget == NULL ||
- !_gtk_widget_is_drawable (self->widget) ||
- _gtk_widget_get_alloc_needed (self->widget))
- return;
-
- if (self->loop_tracker >= 5)
- return;
- self->loop_tracker++;
-
- /* need to clip because widgets may draw out of bounds */
- gtk_snapshot_push_clip (snapshot,
- &GRAPHENE_RECT_INIT(0, 0, width, height));
- graphene_matrix_init_scale (&transform,
- width / gtk_widget_get_allocated_width (self->widget),
- height / gtk_widget_get_allocated_height (self->widget),
- 1.0);
- gtk_snapshot_push_transform (snapshot, &transform);
-
- gtk_widget_snapshot (self->widget, snapshot);
-
- gtk_snapshot_pop (snapshot);
- gtk_snapshot_pop (snapshot);
-
- self->loop_tracker--;
+ gdk_paintable_snapshot (self->current_image, snapshot, width, height);
}
static GdkPaintable *
gtk_widget_paintable_paintable_get_current_image (GdkPaintable *paintable)
{
GtkWidgetPaintable *self = GTK_WIDGET_PAINTABLE (paintable);
- GtkSnapshot *snapshot;
- int width, height;
-
- self->contents_invalid = FALSE;
- self->size_invalid = FALSE;
-
- width = gdk_paintable_get_intrinsic_width (paintable);
- height = gdk_paintable_get_intrinsic_width (paintable);
- if (width == 0 || height == 0)
- return gdk_paintable_new_empty (width, height);
-
- snapshot = gtk_snapshot_new ();
- gdk_paintable_snapshot (GDK_PAINTABLE (self),
- snapshot,
- width, height);
- return gtk_snapshot_free_to_paintable (snapshot, &(graphene_size_t) { width, height });
+
+ return g_object_ref (self->current_image);
}
static int
{
GtkWidgetPaintable *self = GTK_WIDGET_PAINTABLE (paintable);
- self->size_invalid = FALSE;
-
- if (self->widget == NULL)
- return 0;
-
- return gtk_widget_get_allocated_width (self->widget);
+ return gdk_paintable_get_intrinsic_width (self->current_image);
}
static int
{
GtkWidgetPaintable *self = GTK_WIDGET_PAINTABLE (paintable);
- self->size_invalid = FALSE;
-
- if (self->widget == NULL)
- return 0;
-
- return gtk_widget_get_allocated_height (self->widget);
+ return gdk_paintable_get_intrinsic_height (self->current_image);
}
static void
G_OBJECT_CLASS (gtk_widget_paintable_parent_class)->dispose (object);
}
+static void
+gtk_widget_paintable_finalize (GObject *object)
+{
+ GtkWidgetPaintable *self = GTK_WIDGET_PAINTABLE (object);
+
+ g_object_unref (self->current_image);
+
+ G_OBJECT_CLASS (gtk_widget_paintable_parent_class)->finalize (object);
+}
+
static void
gtk_widget_paintable_class_init (GtkWidgetPaintableClass *klass)
{
gobject_class->get_property = gtk_widget_paintable_get_property;
gobject_class->set_property = gtk_widget_paintable_set_property;
gobject_class->dispose = gtk_widget_paintable_dispose;
+ gobject_class->finalize = gtk_widget_paintable_finalize;
/**
* GtkWidgetPaintable:widget
static void
gtk_widget_paintable_init (GtkWidgetPaintable *self)
{
+ self->current_image = gdk_paintable_new_empty (0, 0);
}
/**
NULL);
}
+static GdkPaintable *
+gtk_widget_paintable_snapshot_widget (GtkWidgetPaintable *self)
+{
+ graphene_rect_t bounds;
+
+ if (self->widget == NULL)
+ return gdk_paintable_new_empty (0, 0);
+
+ gtk_widget_compute_bounds (self->widget, self->widget, &bounds);
+
+ if (self->widget->priv->render_node == NULL)
+ return gdk_paintable_new_empty (bounds.size.width, bounds.size.height);
+
+ return gtk_render_node_paintable_new (self->widget->priv->render_node, &bounds);
+}
+
/**
* gtk_widget_paintable_get_widget:
* @self: a #GtkWidgetPaintable
self);
}
+ g_object_unref (self->current_image);
+ self->current_image = gtk_widget_paintable_snapshot_widget (self);
+ g_clear_object (&self->pending_image);
+ if (self->pending_update_cb)
+ {
+ g_source_remove (self->pending_update_cb);
+ self->pending_update_cb = 0;
+ }
+
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_WIDGET]);
gdk_paintable_invalidate_size (GDK_PAINTABLE (self));
gdk_paintable_invalidate_contents (GDK_PAINTABLE (self));
}
-void
-gtk_widget_paintable_invalidate_size (GtkWidgetPaintable *self)
+static gboolean
+gtk_widget_paintable_update_func (gpointer data)
{
- if (self->size_invalid)
- return;
+ GtkWidgetPaintable *self = data;
+ GdkPaintable *old_image;
- self->size_invalid = TRUE;
- gdk_paintable_invalidate_size (GDK_PAINTABLE (self));
+ if (self->current_image != self->pending_image)
+ {
+ old_image = self->current_image;
+ self->current_image = self->pending_image;
+ self->pending_image = NULL;
+ self->pending_update_cb = 0;
+
+ if (gdk_paintable_get_intrinsic_width (self->current_image) != gdk_paintable_get_intrinsic_width (old_image) ||
+ gdk_paintable_get_intrinsic_height (self->current_image) != gdk_paintable_get_intrinsic_height (old_image))
+ gdk_paintable_invalidate_size (GDK_PAINTABLE (self));
+
+ g_object_unref (old_image);
+
+ gdk_paintable_invalidate_contents (GDK_PAINTABLE (self));
+ }
+ else
+ {
+ g_clear_object (&self->pending_image);
+ self->pending_update_cb = 0;
+ }
+
+ return G_SOURCE_REMOVE;
}
void
-gtk_widget_paintable_invalidate_contents (GtkWidgetPaintable *self)
+gtk_widget_paintable_update_image (GtkWidgetPaintable *self)
{
- if (self->contents_invalid)
- return;
+ GdkPaintable *pending_image;
- self->contents_invalid = TRUE;
- gdk_paintable_invalidate_contents (GDK_PAINTABLE (self));
+ if (self->pending_update_cb == 0)
+ {
+ self->pending_update_cb = g_idle_add_full (G_PRIORITY_HIGH,
+ gtk_widget_paintable_update_func,
+ self,
+ NULL);
+ }
+
+ pending_image = gtk_widget_paintable_snapshot_widget (self);
+ g_set_object (&self->pending_image, pending_image);
+ g_object_unref (pending_image);
}